ReactJS 实践心得 key 属性的原理和用法

ReactJS && ReactNative :

本章主要讲解key属性的原理和用法

首先你需要知道:

React与ReactJS && React Native

React是由ReactJSReact Native组成

其中ReactJS是Facebook开源的一个前端框架

React NativeReactJS思想在native上的体现!

既然学过React Native,那你么reactJS呢?

这已经是一个非常流行的框架,其实作为react-native入门,了解一些就够用了。

那么JSX呢?

JSX并不是一门新的语言,仅仅是个语法糖,允许开发者在JavaScript中书写HTML语法。
最后每个HTML标签都转化为JavaScript代码来运行

Start 今天的正题

我们知道,React 元素可以具有一个特殊的属性 key,这个属性不是给用户自己用的,而是给 React 自己用的。

如果我们动态地创建 React 元素,而且 React 元素内包含数量或顺序不确定的子元素时,我们就需要提供 key 这个特殊的属性。

如果你有下面这样的代码:

1
2
3
4
5
6
7
8
9
const UserList = props => (
<div>
<h3>用户列表</h3>
{props.users.map(
u => <div>{u.id}:{u.name}</div>)
}
// 没有提供 key
</div>
)

React 会在控制台打印出报警信息:

1
Warning: Each child in an array or iterator should have a unique "key" prop. Check the render method of `App`. See https://fb.me/react-warning-keys for more information.


你必须为数组中的元素提供唯一的 key 属性,就像下面这样:

1
2
3
4
5
6
7
const UserList = props => (
<div>
<h3>用户列表</h3>
{props.users.map(u => <div key={u.id}>{u.id}:{u.name}</div>)}
// 提供了 key
</div>
)

为什么呢?

我们知道当组件的属性发生了变化,其 render 方法会被重新调用,组件会被重新渲染.
比如:

UserList组件的users属性改变了,就得重新渲染UserList组件,
包括外部的<div>(容器),内部的一个<h3>和若干个<div>(每一个描述一个用户)。

对后一种 <div>(表示用户的),由于其处在一个长度不确定的数组中,
React 需要判断,对数组中的每一项,到底是新建一个元素加入到页面中,还是更新原来的元素。

比如以下几种情况:

[{name: '张三', age: 20}] => [{name: '张三', age: 21}]

这种情况明显只需要更新元素,没有必要重新创建元素。

因为人还是那个人,除了 age,其他信息没有变,显示用户姓名的那个(更小的)元素,是不需要更新(被 ReactDOM 操作到)的。

[{name: '张三'}] => [{name: '张三'}, {name: '李四'}] 

这种情况,显然需要添加一个新元素来表示李四,这个新元素对应的 DOM 元素会被插入到页面中。

[{name: '张三'}] => [{name: '李四'}]

这种情况就有点复杂了,似乎两种方案都可以。
可以把表示张三的元素删掉,为李四新建一个,当然是非常合理的选择。
但是直接把张三的元素换成李四,似乎也无不可。

实际上,如果真的认为上述第3种的后一种方案也无不可,那可是大错特错了。为什么呢:
考虑这种情况:

[{name: '张三'}, {name: '李四'}] => [{name: '李四'}, {name: '张三'}]

难道也需要把张三的元素更新成李四的,李四的元素更新成张三的吗?

那么,为数组中的元素传一个唯一的 key(比如用户的 ID),就很好地解决了这个问题。

React 比较更新前后的元素 key 值,如果相同则更新,如果不同则销毁之前的,重新创建一个元素。

那么,为什么只有数组中的元素需要有唯一的 key,而其他的元素(比如上面的<h3>用户列表</h3>)则不需要呢?

答案是:React 有能力辨别出,更新前后元素的对应关系。

这一点,也许直接看 JSX 不够明显,看 Babel 转换后的 React.createElement 则清晰很多:

转换前:

1
2
3
4
5
6
const element = (
<div>
<h3>example</h3>
{[<p key={1}>hello</p>, <p key={2}>world</p>]}
</div>
)

转换后

1
2
3
4
5
6
7
8
9
10
"use strict";
var element = React.createElement(
"div",
null,
React.createElement("h3",null,"example"),
[
React.createElement("p",{ key: 1 },"hello"),
React.createElement("p",{ key: 2 },"world")
]
)

不管 props 如何变化,数组外的每个元素始终出现在 React.createElement() 参数列表中的固定位置,这个位置就是天然的 key。

题外话:

初学 React 时还容易产生另一个困惑;

那就是为什么 JSX 不支持 if 表达式来有选择地输出
例如:

{if(yes){ <div {...props}/>}})

而必须采用三元运算符来完成这项工作
必须这样:

{yes ? <div {...props}/>} : null)

那是因为,React 需要一个 null 去占住那个元素本来的位置。

吐个槽:

曾经,我天真的以为 key 这个元素只应在数组中使用,直到我在一个复杂的项目中写出了及其恶心的 componentWillReceiveProps方法。我尝试寻找销毁和重建组件,触发componentDidMount* 方法,重置 state,然后才突然发现 key 这个属性已经在那里了。

举个例子:

我们有一个展示用户信息的UserDashboard组件。
传给组件的props只有用户的 基本信息(ID,姓名等),而有关用户的详细信息(比如当前是否在线等)是需要请求过来的。
组件内的一些操作(比如尝试与该用户聊天)也会使用请求,组件本身也有各种状态(比如是否显示聊天框)。

整个界面上最多只有一个UserDashboard,但某些操作(比如点击旁边的 UserList)可能会切换 UserDashboard 的目标用户。

那么问题就来了:

学挖掘机技术哪家强。。。。。咳咳咳串错场子了

当目标用户切换的时候,UserDashboard 仅仅是一个普通的更新操作,触发的是 componentWillReceiveProps,shouldComponentUpdate,componentWillUpdatecomponentDidUpdate 这一套方法。

我们需要在 componentWillReceiveProps 中做太多的事情:

如果我们还不幸地用的 ref 做了一些神奇的 hack,那么你还要去手动把之前做的事情复原回来,这简直要成一团乱麻了!
当 UserDashborad 的逻辑,你的componentWillReceiveProps方法里会充斥着晦涩难懂的只有你能看懂的代码(两周后你自己也看不懂了)。

解决方案是什么?

就是用 key 属性。在 JSX 中使用UserDashboard的时候,不仅把userInfo传入,把 userInfo.id 作为名为 key 的 props 传入(尽管 UserDashboard 不是数组中的组件)。

这样切换目标用户的时候,key 属性也变了,React 会自动销毁之前的组件,用一个全新的组件来渲染新的用户:

我们可以从容地在componentWillUnmount里作清理工作(注销请求的响应函数,防止其更新一个 unmounted component),至于重置 state 这些工作已经不需要做了,由于组件不再是更新,而是销毁和重建,已经是天然完成的。

当然,你可以质疑这样做是否会影响性能。
我认为,只要目标用户的切换不够频繁,对性能的影响是很小的。
如果不使用 key 触发组件的销毁和重建,任由组件自行[更新],每次切换时更新的内容也是很多的。

这时,我们使用 key 带来的性能损耗是完全可以接受的,而带来的收益却非常大。

所以,我想说的结论是:为了组件内部逻辑的清晰,你几乎应该在任何复杂的有状态组件(尤其是有具体对应对象的)上使用key属性(只要 key 属性的改变不是很频繁),这样做,才能在合适的时候触发组件的销毁与重建,组件才能有一个健康的生命周期

又是一个题外话

配合 react-router 时,通常要为 route 组件赋 key,但通常情况下我们是没法传 props 给 route 组件的。

解决的方案是 createElement 方法,如下所示。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
class App extends Component {
static createElement = (Component, ownProps) => {
const {userId} = ownProps.params;
switch (Component) {
case UserDashboard:
return <Component key={userId} {...ownProps}/>;
default:
return <Component {...ownProps}/>;
}
};
render() {
return (
<Provider store={store}>
<Router createElement={App.createElement}
history={syncHistoryWithStore(hashHistory, store)}>
<Route path="/" component={Home}>
<IndexRoute component={Index}/>
<Route path="users/:userId"
component={UserDashboard}/>
</Route>
</Router>
</Provider>
)
}
}

欢迎你的加入!

公众号:Domeday
推送时间为:

AM 7:00 ~ AM 8:30
PM 9:30 ~ PM 11:00

在互联网这个行业,技术的更新迭代速度很快,唯有不断学习和尝试,我们才能立于不败之地,人都是做自己原本不能胜任的事情中,才能快速成长。所以,不要让任何事情成为你不去学习的理由!,你学过的每一样东西,都会在你一生的某个时候派上用场的。

1
GitHub=> React Native BBS 组件已更新